home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
AmigActive 10
/
AACD 10.iso
/
AACD
/
Games
/
MAME
/
src
/
vidhrdw
/
segar.c
< prev
next >
Wrap
C/C++ Source or Header
|
2000-04-23
|
24KB
|
824 lines
/***************************************************************************
vidhrdw.c
Functions to emulate the video hardware of the machine.
***************************************************************************/
#include "driver.h"
#include "vidhrdw/generic.h"
unsigned char *segar_characterram;
unsigned char *segar_characterram2;
unsigned char *segar_mem_colortable;
unsigned char *segar_mem_bcolortable;
typedef struct
{
unsigned char dirtychar[256]; // graphics defined in RAM, mark when changed
unsigned char colorRAM[0x40]; // stored in a 93419 (vid board)
unsigned char bcolorRAM[0x40]; // stored in a 93419 (background board)
unsigned char color_write_enable; // write-enable the 93419 (vid board)
unsigned char flip; // cocktail flip mode (vid board)
unsigned char bflip; // cocktail flip mode (background board)
unsigned char refresh; // refresh the screen
unsigned char brefresh; // refresh the background
unsigned char char_refresh; // refresh the character graphics
unsigned char has_bcolorRAM; // do we have background color RAM?
unsigned char background_enable; // draw the background?
unsigned int back_scene;
unsigned int back_charset;
// used for Pig Newton
unsigned int bcolor_offset;
// used for Space Odyssey
unsigned char backfill;
unsigned char fill_background;
unsigned int backshift;
struct osd_bitmap *horizbackbitmap;
struct osd_bitmap *vertbackbitmap;
} SEGAR_VID_STRUCT;
static SEGAR_VID_STRUCT sv;
/***************************************************************************
The Sega raster games don't have a color PROM. Instead, it has a color RAM
that can be filled with bytes of the form BBGGGRRR. We'll still build up
an initial palette, and set our colortable to point to a different color
for each entry in the colortable, which we'll adjust later using
palette_change_color.
***************************************************************************/
void segar_init_colors(unsigned char *palette, unsigned short *colortable, const unsigned char *color_prom)
{
static unsigned char color_scale[] = {0x00, 0x40, 0x80, 0xC0 };
int i;
/* Our first color needs to be black (transparent) */
*(palette++) = 0;
*(palette++) = 0;
*(palette++) = 0;
/* Space Odyssey uses a static palette for the background, so
our choice of colors isn't exactly arbitrary. S.O. uses a
6-bit color setup, so we make sure that every 0x40 colors
gets a nice 6-bit palette.
(All of the other G80 games overwrite the default colors on startup)
*/
for (i = 0;i < (Machine->drv->total_colors - 1);i++)
{
*(palette++) = color_scale[((i & 0x30) >> 4)];
*(palette++) = color_scale[((i & 0x0C) >> 2)];
*(palette++) = color_scale[((i & 0x03) << 0)];
}
for (i = 0;i < Machine->drv->total_colors;i++)
colortable[i] = i;
}
/***************************************************************************
The two bit planes are separated in memory. If either bit plane changes,
mark the character as modified.
***************************************************************************/
WRITE_HANDLER( segar_characterram_w )
{
sv.dirtychar[offset / 8] = 1;
segar_characterram[offset] = data;
}
WRITE_HANDLER( segar_characterram2_w )
{
sv.dirtychar[offset / 8] = 1;
segar_characterram2[offset] = data;
}
/***************************************************************************
The video port is not entirely understood.
D0 = FLIP
D1 = Color Write Enable (vid board)
D2 = ??? (we seem to need a char_refresh when this is set)
D3 = ??? (looks to be unused on the schems)
D4-D7 = unused?
***************************************************************************/
WRITE_HANDLER( segar_video_port_w )
{
logerror("VPort = %02X\n",data);
if ((data & 0x01) != sv.flip)
{
sv.flip=data & 0x01;
sv.refresh=1;
}
if (data & 0x02)
sv.color_write_enable=1;
else
sv.color_write_enable=0;
if (data & 0x04)
sv.char_refresh=1;
}
/***************************************************************************
If a color changes, refresh the entire screen because it's possible that the
color change affected the transparency (switched either to or from black)
***************************************************************************/
WRITE_HANDLER( segar_colortable_w )
{
static unsigned char red[] = {0x00, 0x24, 0x49, 0x6D, 0x92, 0xB6, 0xDB, 0xFF };
static unsigned char grn[] = {0x00, 0x24, 0x49, 0x6D, 0x92, 0xB6, 0xDB, 0xFF };
static unsigned char blu[] = {0x00, 0x55, 0xAA, 0xFF };
if (sv.color_write_enable)
{
int r,g,b;
b = blu[(data & 0xC0) >> 6];
g = grn[(data & 0x38) >> 3];
r = red[(data & 0x07)];
palette_change_color(offset+1,r,g,b);
if (data == 0)
Machine->gfx[0]->colortable[offset] = Machine->pens[0];
else
Machine->gfx[0]->colortable[offset] = Machine->pens[offset+1];
// refresh the screen if the color switched to or from black
if (sv.colorRAM[offset] != data)
{
if ((sv.colorRAM[offset] == 0) || (data == 0))
{
sv.refresh = 1;
}
}
sv.colorRAM[offset] = data;
}
else
{
logerror("color %02X:%02X (write=%d)\n",offset,data,sv.color_write_enable);
segar_mem_colortable[offset] = data;
}
}
WRITE_HANDLER( segar_bcolortable_w )
{
static unsigned char red[] = {0x00, 0x24, 0x49, 0x6D, 0x92, 0xB6, 0xDB, 0xFF };
static unsigned char grn[] = {0x00, 0x24, 0x49, 0x6D, 0x92, 0xB6, 0xDB, 0xFF };
static unsigned char blu[] = {0x00, 0x55, 0xAA, 0xFF };
int r,g,b;
if (sv.has_bcolorRAM)
{
sv.bcolorRAM[offset] = data;
b = blu[(data & 0xC0) >> 6];
g = grn[(data & 0x38) >> 3];
r = red[(data & 0x07)];
palette_change_color(offset+0x40+1,r,g,b);
}
// Needed to pass the self-tests
segar_mem_bcolortable[offset] = data;
}
/***************************************************************************
Draw the game screen in the given osd_bitmap.
Do NOT call osd_update_display() from this function, it will be called by
the main emulation engine.
***************************************************************************/
int segar_vh_start(void)
{
if (generic_vh_start()!=0)
return 1;
// Init our vid struct, everything defaults to 0
memset(&sv, 0, sizeof(SEGAR_VID_STRUCT));
return 0;
}
/***************************************************************************
This is the refresh code that is common across all the G80 games. This
corresponds to the VIDEO I board.
***************************************************************************/
static void segar_common_screenrefresh(struct osd_bitmap *bitmap, int sprite_transparency, int copy_transparency)
{
int offs;
int charcode;
/* for every character in the Video RAM, check if it has been modified */
/* since last time and update it accordingly. */
for (offs = videoram_size - 1;offs >= 0;offs--)
{
if ((sv.char_refresh) && (sv.dirtychar[videoram[offs]]))
dirtybuffer[offs]=1;
/* Redraw every character if our palette or scene changed */
if ((dirtybuffer[offs]) || sv.refresh)
{
int sx,sy;
sx = 8 * (offs % 32);
sy = 8 * (offs / 32);
if (sv.flip)
{
sx = 31*8 - sx;
sy = 27*8 - sy;
}
charcode = videoram[offs];
/* decode modified characters */
if (sv.dirtychar[charcode] == 1)
{
decodechar(Machine->gfx[0],charcode,segar_characterram,
Machine->drv->gfxdecodeinfo[0].gfxlayout);
sv.dirtychar[charcode] = 2;
}
drawgfx(tmpbitmap,Machine->gfx[0],
charcode,charcode>>4,
sv.flip,sv.flip,sx,sy,
&Machine->drv->visible_area,sprite_transparency,0);
dirtybuffer[offs] = 0;
}
}
for (offs=0;offs<256;offs++)
if (sv.dirtychar[offs]==2)
sv.dirtychar[offs]=0;
/* copy the character mapped graphics */
copybitmap(bitmap,tmpbitmap,0,0,0,0,&Machine->drv->visible_area,copy_transparency,Machine->pens[0]);
sv.char_refresh=0;
sv.refresh=0;
}
/***************************************************************************
"Standard" refresh for games without special background boards.
***************************************************************************/
void segar_vh_screenrefresh(struct osd_bitmap *bitmap,int full_refresh)
{
if (palette_recalc() || full_refresh)
sv.refresh = 1;
segar_common_screenrefresh(bitmap, TRANSPARENCY_NONE, TRANSPARENCY_NONE);
}
/***************************************************************************
---------------------------------------------------------------------------
Space Odyssey Functions
---------------------------------------------------------------------------
***************************************************************************/
/***************************************************************************
Create two background bitmaps for Space Odyssey - one for the horizontal
scrolls that's 4 times wider than the screen, and one for the vertical
scrolls that's 4 times taller than the screen.
***************************************************************************/
int spaceod_vh_start(void)
{
if (segar_vh_start()!=0)
return 1;
if ((sv.horizbackbitmap = osd_create_bitmap(4*Machine->drv->screen_width,Machine->drv->screen_height)) == 0)
{
generic_vh_stop();
return 1;
}
if ((sv.vertbackbitmap = osd_create_bitmap(Machine->drv->screen_width,4*Machine->drv->screen_height)) == 0)
{
osd_free_bitmap(sv.horizbackbitmap);
generic_vh_stop();
return 1;
}
return 0;
}
/***************************************************************************
Get rid of the Space Odyssey background bitmaps.
***************************************************************************/
void spaceod_vh_stop(void)
{
osd_free_bitmap(sv.horizbackbitmap);
osd_free_bitmap(sv.vertbackbitmap);
generic_vh_stop();
}
/***************************************************************************
This port controls which background to draw for Space Odyssey. The temp_scene
and temp_charset are analogous to control lines used to select the background.
If the background changed, refresh the screen.
***************************************************************************/
WRITE_HANDLER( spaceod_back_port_w )
{
unsigned int temp_scene, temp_charset;
temp_scene = (data & 0xC0) >> 6;
temp_charset = (data & 0x04) >> 2;
if (temp_scene != sv.back_scene)
{
sv.back_scene=temp_scene;
sv.brefresh=1;
}
if (temp_charset != sv.back_charset)
{
sv.back_charset=temp_charset;
sv.brefresh=1;
}
/* Our cocktail flip-the-screen bit. */
if ((data & 0x01) != sv.bflip)
{
sv.bflip=data & 0x01;
sv.brefresh=1;
}
sv.background_enable=1;
sv.fill_background=0;
}
/***************************************************************************
This port controls the Space Odyssey background scrolling. Each write to
this port scrolls the background by one bit. Faster speeds are achieved
by the program writing more often to this port. Oddly enough, the value
sent to this port also seems to indicate the speed, but the value itself
is never checked.
***************************************************************************/
WRITE_HANDLER( spaceod_backshift_w )
{
sv.backshift= (sv.backshift + 1) % 0x400;
sv.background_enable=1;
sv.fill_background=0;
}
/***************************************************************************
This port resets the Space Odyssey background to the "top". This is only
really important for the Black Hole level, since the only way the program
can line up the background's Black Hole with knowing when to spin the ship
is to force the background to restart every time you die.
***************************************************************************/
WRITE_HANDLER( spaceod_backshift_clear_w )
{
sv.backshift=0;
sv.background_enable=1;
sv.fill_background=0;
}
/***************************************************************************
Space Odyssey also lets you fill the background with a specific color.
***************************************************************************/
WRITE_HANDLER( spaceod_backfill_w )
{
sv.backfill=data + 0x40 + 1;
sv.fill_background=1;
}
/***************************************************************************
***************************************************************************/
WRITE_HANDLER( spaceod_nobackfill_w )
{
sv.backfill=0;
sv.fill_background=0;
}
/***************************************************************************
Special refresh for Space Odyssey, this code refreshes the static background.
***************************************************************************/
void spaceod_vh_screenrefresh(struct osd_bitmap *bitmap,int full_refresh)
{
int offs;
int charcode;
int sprite_transparency;
int vert_scene;
unsigned char *back_charmap = memory_region(REGION_USER1);
if (palette_recalc() || full_refresh)
sv.refresh = 1;
// scenes 0,1 are horiz. scenes 2,3 are vert.
vert_scene = !(sv.back_scene & 0x02);
sprite_transparency=TRANSPARENCY_PEN;
/* If the background picture changed, draw the new one in temp storage */
if (sv.brefresh)
{
sv.brefresh=0;
for (offs = 0x1000 - 1; offs >= 0; offs--)
{
int sx,sy;
/* Use Vertical Back Scene */
if (vert_scene)
{
sx = 8 * (offs % 32);
sy = 8 * (offs / 32);
if (sv.bflip)
{
sx = 31*8 - sx;
sy = 127*8 - sy;
}
}
/* Use Horizontal Back Scene */
else
{
sx = (8 * (offs % 32)) + (256 * (offs >> 10));
sy = 8 * ((offs & 0x3FF) / 32);
if (sv.bflip)
{
sx = 127*8 - sx;
sy = 31*8 - sy; /* is this right? */
}
}
charcode = back_charmap[(sv.back_scene*0x1000) + offs];
if (vert_scene)
{
drawgfx(sv.vertbackbitmap,Machine->gfx[1 + sv.back_charset],
charcode,0,
sv.bflip,sv.bflip,sx,sy,
0,TRANSPARENCY_NONE,0);
}
else
{
drawgfx(sv.horizbackbitmap,Machine->gfx[1 + sv.back_charset],
charcode,0,
sv.bflip,sv.bflip,sx,sy,
0,TRANSPARENCY_NONE,0);
}
}
}
/* Copy the scrolling background */
{
int scrollx,scrolly;
if (vert_scene)
{
if (sv.bflip)
scrolly = sv.backshift;
else
scrolly = -sv.backshift;
copyscrollbitmap(bitmap,sv.vertbackbitmap,0,0,1,&scrolly,&Machine->drv->visible_area,TRANSPARENCY_NONE,0);
}
else
{
if (sv.bflip)
scrollx = sv.backshift;
else
scrollx = -sv.backshift;
scrolly = -32;
copyscrollbitmap(bitmap,sv.horizbackbitmap,1,&scrollx,1,&scrolly,&Machine->drv->visible_area,TRANSPARENCY_NONE,0);
}
}
if (sv.fill_background==1)
{
fillbitmap(bitmap,Machine->pens[sv.backfill],&Machine->drv->visible_area);
}
/* Refresh the "standard" graphics */
segar_common_screenrefresh(bitmap, TRANSPARENCY_NONE, TRANSPARENCY_PEN);
}
/***************************************************************************
---------------------------------------------------------------------------
Monster Bash Functions
---------------------------------------------------------------------------
***************************************************************************/
int monsterb_vh_start(void)
{
if (segar_vh_start()!=0)
return 1;
sv.has_bcolorRAM = 1;
return 0;
}
/***************************************************************************
This port controls which background to draw for Monster Bash. The tempscene
and tempoffset are analogous to control lines used to bank switch the
background ROMs. If the background changed, refresh the screen.
***************************************************************************/
WRITE_HANDLER( monsterb_back_port_w )
{
unsigned int temp_scene, temp_charset;
temp_scene = 0x400 * ((data & 0x70) >> 4);
temp_charset = data & 0x03;
if (sv.back_scene != temp_scene)
{
sv.back_scene = temp_scene;
sv.refresh=1;
}
if (sv.back_charset != temp_charset)
{
sv.back_charset = temp_charset;
sv.refresh=1;
}
/* This bit turns the background off and on. */
if ((data & 0x80) && (sv.background_enable==0))
{
sv.background_enable=1;
sv.refresh=1;
}
else if (((data & 0x80)==0) && (sv.background_enable==1))
{
sv.background_enable=0;
sv.refresh=1;
}
}
/***************************************************************************
Special refresh for Monster Bash, this code refreshes the static background.
***************************************************************************/
void monsterb_vh_screenrefresh(struct osd_bitmap *bitmap,int full_refresh)
{
int offs;
int charcode;
int sprite_transparency;
unsigned char *back_charmap = memory_region(REGION_USER1);
if (palette_recalc() || full_refresh)
sv.refresh = 1;
sprite_transparency=TRANSPARENCY_NONE;
/* If the background is turned on, refresh it first. */
if (sv.background_enable)
{
/* for every character in the Video RAM, check if it has been modified */
/* since last time and update it accordingly. */
for (offs = videoram_size - 1;offs >= 0;offs--)
{
if ((sv.char_refresh) && (sv.dirtychar[videoram[offs]]))
dirtybuffer[offs]=1;
/* Redraw every background character if our palette or scene changed */
if ((dirtybuffer[offs]) || sv.refresh)
{
int sx,sy;
sx = 8 * (offs % 32);
sy = 8 * (offs / 32);
if (sv.flip)
{
sx = 31*8 - sx;
sy = 27*8 - sy;
}
charcode = back_charmap[offs + sv.back_scene];
drawgfx(tmpbitmap,Machine->gfx[1 + sv.back_charset],
charcode,((charcode & 0xF0)>>4),
sv.flip,sv.flip,sx,sy,
&Machine->drv->visible_area,TRANSPARENCY_NONE,0);
}
}
sprite_transparency=TRANSPARENCY_PEN;
}
/* Refresh the "standard" graphics */
segar_common_screenrefresh(bitmap, sprite_transparency, TRANSPARENCY_NONE);
}
/***************************************************************************
---------------------------------------------------------------------------
Pig Newton Functions
---------------------------------------------------------------------------
***************************************************************************/
/***************************************************************************
This port seems to control the background colors for Pig Newton.
***************************************************************************/
WRITE_HANDLER( pignewt_back_color_w )
{
if (offset == 0)
{
sv.bcolor_offset = data;
}
else
{
segar_bcolortable_w(sv.bcolor_offset,data);
}
}
/***************************************************************************
These ports control which background to draw for Pig Newton. They might
also control other video aspects, since without schematics the usage of
many of the data lines is indeterminate. Segar_back_scene and segar_backoffset
are analogous to registers used to control bank-switching of the background
"videorom" ROMs and the background graphics ROMs, respectively.
If the background changed, refresh the screen.
***************************************************************************/
WRITE_HANDLER( pignewt_back_ports_w )
{
unsigned int tempscene;
logerror("Port %02X:%02X\n",offset + 0xb8,data);
/* These are all guesses. There are some bits still being ignored! */
switch (offset)
{
case 1:
/* Bit D7 turns the background off and on? */
if ((data & 0x80) && (sv.background_enable==0))
{
sv.background_enable=1;
sv.refresh=1;
}
else if (((data & 0x80)==0) && (sv.background_enable==1))
{
sv.background_enable=0;
sv.refresh=1;
}
/* Bits D0-D1 help select the background? */
tempscene = (sv.back_scene & 0x0C) | (data & 0x03);
if (sv.back_scene != tempscene)
{
sv.back_scene = tempscene;
sv.refresh=1;
}
break;
case 3:
/* Bits D0-D1 help select the background? */
tempscene = ((data << 2) & 0x0C) | (sv.back_scene & 0x03);
if (sv.back_scene != tempscene)
{
sv.back_scene = tempscene;
sv.refresh=1;
}
break;
case 4:
if (sv.back_charset != (data & 0x03))
{
sv.back_charset = data & 0x03;
sv.refresh=1;
}
break;
}
}
/***************************************************************************
---------------------------------------------------------------------------
Sinbad Mystery Functions
---------------------------------------------------------------------------
***************************************************************************/
/***************************************************************************
Controls the background image
***************************************************************************/
WRITE_HANDLER( sindbadm_back_port_w )
{
unsigned int tempscene;
/* Bit D7 turns the background off and on? */
if ((data & 0x80) && (sv.background_enable==0))
{
sv.background_enable=1;
sv.refresh=1;
}
else if (((data & 0x80)==0) && (sv.background_enable==1))
{
sv.background_enable=0;
sv.refresh=1;
}
/* Bits D2-D6 select the background? */
tempscene = (data >> 2) & 0x1F;
if (sv.back_scene != tempscene)
{
sv.back_scene = tempscene;
sv.refresh=1;
}
/* Bits D0-D1 select the background char set? */
if (sv.back_charset != (data & 0x03))
{
sv.back_charset = data & 0x03;
sv.refresh=1;
}
}
/***************************************************************************
Special refresh for Sinbad Mystery, this code refreshes the static background.
***************************************************************************/
void sindbadm_vh_screenrefresh(struct osd_bitmap *bitmap,int full_refresh)
{
int offs;
int charcode;
int sprite_transparency;
unsigned long backoffs;
unsigned long back_scene;
unsigned char *back_charmap = memory_region(REGION_USER1);
if (palette_recalc() || full_refresh)
sv.refresh = 1;
sprite_transparency=TRANSPARENCY_NONE;
/* If the background is turned on, refresh it first. */
if (sv.background_enable)
{
/* for every character in the Video RAM, check if it has been modified */
/* since last time and update it accordingly. */
for (offs = videoram_size - 1;offs >= 0;offs--)
{
if ((sv.char_refresh) && (sv.dirtychar[videoram[offs]]))
dirtybuffer[offs]=1;
/* Redraw every background character if our palette or scene changed */
if ((dirtybuffer[offs]) || sv.refresh)
{
int sx,sy;
sx = 8 * (offs % 32);
sy = 8 * (offs / 32);
if (sv.flip)
{
sx = 31*8 - sx;
sy = 27*8 - sy;
}
// NOTE: Pig Newton has 16 backgrounds, Sinbad Mystery has 32
back_scene = (sv.back_scene & 0x1C) << 10;
backoffs = (offs & 0x01F) + ((offs & 0x3E0) << 2) + ((sv.back_scene & 0x03) << 5);
charcode = back_charmap[backoffs + back_scene];
drawgfx(tmpbitmap,Machine->gfx[1 + sv.back_charset],
charcode,((charcode & 0xF0)>>4),
sv.flip,sv.flip,sx,sy,
&Machine->drv->visible_area,TRANSPARENCY_NONE,0);
}
}
sprite_transparency=TRANSPARENCY_PEN;
}
/* Refresh the "standard" graphics */
segar_common_screenrefresh(bitmap, sprite_transparency, TRANSPARENCY_NONE);
}